wndpanel.js at [fdb5b85112] Вы: nobody
Вход

File wndpanel.js as of check-in [fdb5b85112]


$engine JScript
$uname wndpanel
$dname Панель окон
$addin vbs
$addin global
$addin stdlib
$addin stdcommands

// (c) Александр Орефков
// Скрипт для показа "панели окон".
// В отличии от штатной панели окон показывает список окон в табличном поле, сортируя
// их не в порядке открытия окон, а по объектам метаданных, к которым они относятся +
// по алфавиту. Также как всегда поддерживается фильтрация списка по подстроке.

// Переработка для показа в дереве: Пушин Владимир <vladnet@gmail.com>

global.connectGlobals(SelfScript)


var form
var needActivate, needHide
var api = stdlib.require('winapi.js')

function getFullMDName(mdObj, mdProp)
{
    var names = []
    while(true)
    {
        names.unshift(mdObj.name)
        var className = mdObj.mdclass.name(1)
        if(!mdObj.parent)
            className = ""
        names.unshift(className)
        if(!className.length)
            break
        mdObj = mdObj.parent
    }
    return names.join('.') + "#" + (mdProp ? mdProp.name(1) : "")
}

WndListItem = stdlib.Class.extend(
{
    construct: function(view)
    {
        this.view = view
        this.rowInVt = null
        this.color = 0
        this.makeSortKey()
    },
    isAlive: function()
    {
        try{
            if(this.view.hwnd && this.view.position().state == vsMDI)
                return true
        }catch(e){}
        return false
    },
    makeTitle: function()
    {
        var result = {title : '', info: ''}
        if(this.isAlive())
        {
            result.title = this.view.title
            var mdObj = this.view.mdObj
            if(mdObj)
            {
                var mdname = mdObj.container.identifier
                if(result.title.indexOf(mdname) < 0)
                    result.info += mdname + " "
            }
            var obj = this.view.getObject()
            if(obj)
                result.info += toV8Value(obj).typeName(1) + " "
        }
        return result
    },
    makeSortKey : function()
    {
        // Основной алгоритм упорядочивания окон
        var md = this.view.mdObj
        if(md)
        {
            // Если окно относится к объекту метаданных. Сначала пусть идут окна
            // основной конфигурации, далее конфигурации ИБ, затем внешние отчеты/обработки и cf-ники.
            // При закрытой основной конфигурации metadata.current равно metadata.ib, поэтому сначала
            // проверяем на metadata.ib
            if(md.container == metadata.ib)
                this.sortkey = "2#"
            else if(md.container == metadata.current)
                this.sortkey = "1#"
            else
                this.sortkey = "3#" + md.container.identifier + "#"
            this.sortkey += getFullMDName(md, this.view.mdProp)
        }
        else    // Дальше пусть идут всякие файлы по алфавиту
            this.sortkey = "4#" + this.view.title
        this.sortkey = this.sortkey.toLowerCase()
    }
})

WndList = stdlib.Class.extend({
    construct: function()
    {
        this.list = []  // Массив - список окон
        this.find = {}  // Для поиска окна по его id
        this.lastFilter = ''
        this.activeView = null
    },
    // Функция для удаления устаревших, закрытых окон из нашего списка
    removeOldViews: function(vt)
    {
        var removed = false
        for(var i = this.list.length; i--;)
        {
            var item = this.list[i]
            if(!item.isAlive())
            {
		        try{ // попытаемся получить Родителя если не сможем значит строки уже нет
		            var test=item.rowInVt.Родитель
		        }catch(e){
		        	return true
		        }
                if(item.rowInVt)
                {
                	if(item.rowInVt.Родитель == undefined)
                		vt.Rows.Delete(item.rowInVt)
                	else
                		item.rowInVt.Родитель.Rows.Delete(item.rowInVt)
                }
                
                delete this.find[item.view.id]
                this.list.splice(i, 1)
                removed = true
            }
        }
        return removed
    },
    // Функция для добавления новых окон в список.
    // Перебирает все MDI-окна, и те, которых нет в списке, добавляет туда
    // Также определяет активное окно
    addNewViews: function()
    {
        var views = []      // Массив всех конечных отображений
        var childs = windows.mdiView.enumChilds();   // Получим список MDI-окон
        (function(views, list)  // Далее надо каждое MDI-окно "раскрутить" до конечных отображений,
        {                       // т.к. MDI-окно может быть контейнером для одного или нескольких отображений
            for(var i = 0; i < views.count; i++)
            {
                var v = views.item(i)
                if(v.isContainer != vctNo)  // Окно - контейнер. Рекурсивно раскрутим его потомков
                    arguments.callee(v.enumChilds(), list)
                else    // Окно не контейнер. Добавим в общий список
                    list.push(v) 
            }
        })(childs, views)
        var added = false
        // Перебираем весь список окон
        for(var idx in views)
        {
            var v = views[idx]
            if(!this.find.hasOwnProperty(v.id))
            {
                var item = new WndListItem(v)
                this.list.push(item)
                this.find[v.id] = item
                added = true
            }
        }
        if(added)   // Что-то добавилось, отсортируем список
            this.list.sort(function(i1, i2){return i1.sortkey.localeCompare(i2.sortkey)})
        var activeView = null
        if(childs.count > 0)
        {
            activeView = childs.item(0)
            while(activeView.activeChild)
                activeView = activeView.activeChild
            activeView = this.find[activeView.id]
        }
        return {added: added, activeView: activeView}
    },
    filterList: function(filterString, vtControl)
    {
        vt = vtControl.Value
        var needUpdateColors = this.removeOldViews(vt)
        filterString = filterString.toLowerCase()
        var addedResults = this.addNewViews()
        if(addedResults.added || filterString != this.lastFilter)
        {
            needUpdateColors = true            
            this.lastFilter = filterString
            var filters = filterString.split(/\s+/)
            var idxInVt = 0
            for(var vidx in this.list)
            {
                var item = this.list[vidx]
                var needAdd = true
                var title = item.makeTitle().title.toLowerCase()
                for(var idx in filters)
                {
                    if(title.indexOf(filters[idx]) < 0)
                    {
                        needAdd = false
                        break
                    }
                }
                if(needAdd)
                {
                    if(!item.rowInVt)
                    {
                    	лЗаголовок=item.makeTitle().title;
	                   	лПозицияДвоеточия=лЗаголовок.indexOf(': ')
	                   	
    		            if(лПозицияДвоеточия == -1)
            		    {
	                        item.rowInVt = vt.Rows.Insert(idxInVt)
                    	}
		                else 
		                {
		                	лРодитель = vt.Rows.Найти(лЗаголовок.substr(0, лПозицияДвоеточия), "Заголовок", true)
	    		            if(лРодитель == undefined)
		                        item.rowInVt = vt.Rows.Insert(idxInVt)
			                else
		                		item.rowInVt = лРодитель.Rows.Insert(idxInVt)
		                	лЗаголовок = лЗаголовок.substr(лПозицияДвоеточия+1)
		                }
                        
                        item.rowInVt.Окно = item
                        item.rowInVt.Заголовок = лЗаголовок;
                    }
                    idxInVt++
                }
                else if(item.rowInVt)
                {
					try{
			            vt.Rows.Delete(item.rowInVt)
			        }catch(e){}
                    item.rowInVt = null
                }
            }
        }
        if(needUpdateColors && vt.Rows.Count())
        {
            var prevItem = vt.Rows.Get(0).Окно
            prevItem.color = 0
            for(var k = 1; k < vt.Rows.Count(); k++)
            {
                var item = vt.Rows.Get(k).Окно
                item.color = (prevItem.color + 1) % 2
                var mdObj = item.view.mdObj
                var prevMdObj = prevItem.view.mdObj
                if(mdObj && prevMdObj)
                {
                    // Текущая строка - метаданные, и предыдущая строка - метаданные.
                    // Если они относятся к одному объекту, то цвет должен совпадать.
                    if(mdObj.container == prevMdObj.container)  // Находятся в одном контейнере
                    {
                        // Если это - внешняя обработка или принадлежат одному объекту первого уровня
                        if(mdObj.container.masterContainer != mdObj.container || find1LevelMdObj(mdObj) == find1LevelMdObj(prevMdObj))
                            item.color = prevItem.color
                    }
                }
                prevItem = item
            }
        }
        // Теперь отследим активное окно
        oldActiveView = this.activeView
        if(addedResults.activeView != oldActiveView)
        {
            this.activeView = addedResults.activeView
            if(oldActiveView && oldActiveView.rowInVt)
                vtControl.RefreshRows(oldActiveView.rowInVt)
            if(addedResults.activeView && addedResults.activeView.rowInVt)
            {
                vtControl.RefreshRows(addedResults.activeView.rowInVt)
                vtControl.ТекущаяСтрока = addedResults.activeView.rowInVt
            }
        }
    }
})

function macrosПоказать()
{
    form.Filter = ""
    form.Открыть()
    form.CurrentControl = form.Controls.WndList
    if (activateSearchElement){
        form.CurrentControl = form.Controls.Filter;
    }
}

function macrosПереключитьВидимостьОкнаСвойств()
{
    windows.propsVisible = !windows.propsVisible
}

/* Возвращает название макроса по умолчанию - вызывается, когда пользователь 
дважды щелкает мышью по названию скрипта в окне Снегопата. */
function getDefaultMacros() {
    return 'Показать';
}

function updateWndList()
{
    // Получим текущий текст из поля ввода
    vbs.var0 = form.Controls.Filter
    vbs.DoExecute("var0.GetTextSelectionBounds var1, var2, var3, var4")
    form.Controls.Filter.УстановитьГраницыВыделения(1, 1, 1, 10000)
    var newText = form.Controls.Filter.ВыделенныйТекст.replace(/^\s*|\s*$/g, '')
    form.Controls.Filter.УстановитьГраницыВыделения(vbs.var1, vbs.var2, vbs.var3, vbs.var4)
    WndList.One.filterList(newText, form.Controls.WndList)
}

function onIdle()
{
    updateWndList()
    if(needHide)
    {
        needHide = false
        // Теперь спрячем наше окно.
        // Для прячущегося окна нельзя делать form.Close, т.к. тогда оно пропадет совсем, не оставив кнопки на панели
        if(form.СостояниеОкна != ВариантСостоянияОкна.Прячущееся)
            form.Close()
    }
    if(needActivate)
    {
        try{
            needActivate.activate()
        }catch(e){}
        needActivate = null
    }
}

function withSelected(func)
{
    for(var rows = new Enumerator(form.Controls.WndList.ВыделенныеСтроки); !rows.atEnd(); rows.moveNext())
        func(rows.item().Окно)
}

function WndListВыбор(Элемент, ВыбраннаяСтрока, Колонка, СтандартнаяОбработка)
{
    needActivate = ВыбраннаяСтрока.val.Окно.view
    СтандартнаяОбработка.val = false
}

var boldFontV8, fontWin, boldFontWin

function ВыделитьИмяФайлаИзПолногоПути(пПуть, СРасширением)
{
	if(СРасширением)
		var expr=/.*\\([\W\w\-\.]+)/
	else
		var expr=/.*\/([\W\w\-\.]+)\.[^#?\s]+?$/;
	if (пПуть.match(expr))
		return RegExp.$1
	return пПуть
}

function WndListПриВыводеСтроки(Элемент, ОформлениеСтроки, ДанныеСтроки)
{
    var cell = ОформлениеСтроки.val.Ячейки.Окно
    var item = ДанныеСтроки.val.Окно
    try{cell.УстановитьКартинку(item.view.icon)}catch(e){}
    var title = item.makeTitle()
    var hdc = api.GetDC(0)
    
    var titlestr =  title.title
    if(ДанныеСтроки.val.Родитель != undefined)
    	titlestr = ДанныеСтроки.val.Заголовок
    
    // Приготовим шрифты.
    if(!boldFontV8)
    {
        boldFontV8 = v8New("Шрифт", cell.Шрифт, undefined, undefined, true)
        fontWin = api.CreateApiFontFromV8Font(cell.Шрифт, hdc)
        boldFontWin = api.CreateApiFontFromV8Font(boldFontV8, hdc)
    }
    // Рассчет ширины колонок и текста
    // Прямого способа получить ширину колонок в пикселях нет, поэтому рассчитаем ширину колонки "Окно"
    // пропорционально общей ширине в пикселах
    var widthOfColumn = form.Controls.WndList.Ширина * form.Controls.WndList.Колонки.Окно.Ширина /
        (form.Controls.WndList.Колонки.Окно.Ширина + form.Controls.WndList.Колонки.Инфо.Ширина)
        - 50 // Иконка окна и отступы от рамки
    var apiFont = fontWin
    if(item == WndList.One.activeView)
    {
        cell.Шрифт = boldFontV8
        apiFont = boldFontWin
        widthOfColumn -= 20
    }
    ОформлениеСтроки.val.ЦветФона = item.color ?  Элемент.val.ЦветФонаЧередованияСтрок : Элемент.val.ЦветФонаПоля
    
    if(мДляВнешнихФайловОтображатьТолькоИмяФайла)
    {
    	titlestr2=ВыделитьИмяФайлаИзПолногоПути(titlestr, true)
    	ОформлениеСтроки.val.Ячейки.Окно.УстановитьТекст(titlestr2)
    	
    	if(titlestr2 != titlestr)
    		ОформлениеСтроки.val.Ячейки.Инфо.УстановитьТекст("[" + titlestr + "]")
    	else
    		ОформлениеСтроки.val.Ячейки.Инфо.УстановитьТекст(title.info)
    	return
    }
    
    var oldFont = api.SelectObject(hdc, apiFont)
    // без таких ухищрений (гарантированное создание копии строки) переменные oldTitle и title.title
    // будут ссылаться на одну и ту же область памяти со строкой, а так как dynwrapx модифицирует
    // буфер строки напрямую, то oldTitle и title.title всегда будут равны, даже если DrawText
    // изменит строку
    var oldTitle = new String("-" + titlestr)
    var res = api.DrawText(hdc, titlestr,
	    new api.Rect(0, 0, widthOfColumn, 0), 0x20 | 0x4000 | 0x10000 | 0x400)// DT_CALCRECT | DT_SINGLELINE | DT_PATH_ELLIPSIS | DT_MODIFYSTRING
    cell.УстановитьТекст(res.text)  // Если текст был шире колонки, то DrawText изменит его так, чтобы он влезал
    api.SelectObject(hdc, oldFont)
    api.ReleaseDC(0, hdc)
    if("-" + res.text != oldTitle)
        title.info += "[" + oldTitle.substr(1) + "]"
    ОформлениеСтроки.val.Ячейки.Инфо.УстановитьТекст(title.info)
}

function FilterРегулирование(Элемент, Направление, СтандартнаяОбработка)
{
    var curRow = form.Controls.WndList.ТекущаяСтрока;
    var wndList = form.Controls.WndList.Value;
    if(!curRow)
    {
        if(form.WndList.Rows.Количество())
            form.Controls.WndList.ТекущаяСтрока = form.WndList.Rows.Получить(-1 == Направление.val ? 0 : form.WndList.Rows.Количество() - 1)
        return
    }
    var curRowIdx = form.WndList.Rows.Индекс(curRow), newRowIdx = curRowIdx
    
    if(-1 == Направление.val)
    {
        if(curRowIdx != form.WndList.Rows.Количество() - 1)
            newRowIdx++
    }
    else
    {
        if(curRowIdx > 0)
            newRowIdx--
    }
    if(newRowIdx != curRowIdx)
        form.Controls.WndList.ТекущаяСтрока = form.WndList.Rows.Получить(newRowIdx)
    СтандартнаяОбработка.val = false
}

function ПриОткрытии()
{
    updateWndList()
    events.connect(Designer, "onIdle", SelfScript.self)
    form.Controls.Cmds.Кнопки.SaveSession.Доступность = мИспользоватьСессии;
    form.Controls.Cmds.Кнопки.RestoreSession.Доступность = мИспользоватьСессии;
    
}
function ПриЗакрытии()
{
    events.disconnect(Designer, "onIdle", SelfScript.self)
}

function find1LevelMdObj(mdObj)
{
    if(mdObj.mdclass.name(1).length)
    {
        while(mdObj.parent && mdObj.parent.parent)
            mdObj = mdObj.parent
    }
    return mdObj
}

function CmdsActivate(Кнопка)
{
    if(form.Controls.WndList.ТекущаяСтрока)    
        needActivate = form.Controls.WndList.ТекущаяСтрока.Окно.view
}

function closeSelected()
{
    withSelected(function(item){item.view.close()})
}

function CmdsClose(Кнопка)
{
    closeSelected()
}

function CmdsSave(Кнопка)
{
    withSelected(function(item){
        stdcommands.Frame.FileSave.sendToView(item.view)
        form.Controls.WndList.ОбновитьСтроки(item.rowInVt)
    })
}

function CmdsFindInTree(Кнопка)
{
    if(form.Controls.WndList.ТекущаяСтрока)
    {
        var view = form.Controls.WndList.ТекущаяСтрока.Окно.view
        if(view.mdObj)
            view.mdObj.activateInTree()
    }
}

function CmdsMinimizeAll(Кнопка)
{
    var views = windows.mdiView.enumChilds()
    for(var k = 0; k < views.count; k++)
        views.item(k).sendCommand("{c9d3c390-1eb4-11d5-bf52-0050bae2bc79}", 6)
}

function CmdsPrint(Кнопка)
{
    withSelected(function(item){
        stdcommands.Frame.Print.sendToView(item.view)
    })
}

function CmdsSaveSession(Кнопка){

    if (!sessionManager)
        return
    nameSession = sessionManager.choiceSessionName();
    if (!nameSession)
        return;
    var views = {};
    for(var rows = new Enumerator(form.Controls.WndList.ВыделенныеСтроки); !rows.atEnd(); rows.moveNext()) {
        item = rows.item().Окно;
        views[item.view.id] = item;
    }
    sessionManager.saveSession(nameSession, views, 'SessionSaved');

}

function CmdsRestoreSession(Кнопка){

    if (!sessionManager)
        return
    nameSession = sessionManager.choiceSessionName();
    if (!nameSession)
        return;
    sessionManager.restoreSession(nameSession, 'SessionSaved');
    
}

function НастройкиПриОткрытии() {
    мФормаНастройки.ДляВнешнихФайловОтображатьТолькоИмяФайла=мДляВнешнихФайловОтображатьТолькоИмяФайла
    мФормаНастройки.ИспользоватьСессии = мИспользоватьСессии;
    мФормаНастройки.ПриОткрытииФормыАктивизироватьСтрокуПоиска = activateSearchElement;
}

function CmdsConfig(Кнопка)
{
	var pathToForm=SelfScript.fullPath.replace(/.js$/, 'param.ssf')
    мФормаНастройки=loadScriptForm(pathToForm, SelfScript.self) // Обработку событий формы привяжем к самому скрипту
    мФормаНастройки.ОткрытьМодально()
}

function мЗаписатьНастройки() {
    мДляВнешнихФайловОтображатьТолькоИмяФайла=мФормаНастройки.ДляВнешнихФайловОтображатьТолькоИмяФайла
    мИспользоватьСессии = мФормаНастройки.ИспользоватьСессии;
    activateSearchElement = мФормаНастройки.ПриОткрытииФормыАктивизироватьСтрокуПоиска;
    profileRoot.setValue(pflOnlyNameForExtFiles, мДляВнешнихФайловОтображатьТолькоИмяФайла)
    profileRoot.setValue(pflUseSessions, мИспользоватьСессии);
    profileRoot.setValue(pflActivateSearch, activateSearchElement);
    if (!sessionManager && мИспользоватьСессии){
        //Message("test load settings")
        loadSessionManager();
    }
}

function CmdsConfigSaveClose(Кнопка) {
    мЗаписатьНастройки()
    мФормаНастройки.Закрыть()
}

function CmdsConfigSave(Кнопка) {
    мЗаписатьНастройки()
}

function InvisiblePanelSelectAndHide(Кнопка)
{
    if(form.Controls.WndList.ТекущаяСтрока)
    {
        needActivate = form.Controls.WndList.ТекущаяСтрока.Окно.view
        needHide = true
    }
}

function WndListПередНачаломДобавления(Элемент, Отказ, Копирование)
{
    Отказ.val = true
}

function WndListПередУдалением(Элемент, Отказ)
{
    Отказ.val = true
    closeSelected()
}

(function(){
    // Инициализация скрипта
    WndList.One = new WndList
    form = loadScriptForm(SelfScript.fullPath.replace(/js$/, 'ssf'), SelfScript.self)
    form.КлючСохраненияПоложенияОкна = "wndpanel"
    form.WndList.Columns.Окно.ТипЗначения = v8New("ОписаниеТипов")
    var hk = [
    ["Activate", 13, 0],
    ["Close", 115, 8],
    ["Save", "S".charCodeAt(0), 8],
    ["Print", "P".charCodeAt(0), 8],
    ["FindInTree", "T".charCodeAt(0), 8]
    ]
    for(var k in hk)
        form.Controls.Cmds.Кнопки.Найти(hk[k][0]).СочетаниеКлавиш = stdlib.v8hotkey(hk[k][1], hk[k][2])
    form.Controls.InvisiblePanel.Кнопки.SelectAndHide.СочетаниеКлавиш = stdlib.v8hotkey(13,8)
})()

function loadSessionManager(){
    try {
        sessionManager = stdlib.require(stdlib.getSnegopatMainFolder()+"scripts\\SessionManager.js").GetSessionManager();    
    } catch(e){
        Message("Невозможно загрузить Менеджер сессий "+e.description());
    };
}

var pflOnlyNameForExtFiles = "WndPanel/OnlyNameForExtFiles"
var pflUseSessions = "WndPanel/UseSessions";
var pflActivateSearch = "WndPanel/ActivateSearch";
profileRoot.createValue(pflOnlyNameForExtFiles, false, pflSnegopat)
profileRoot.createValue(pflUseSessions, false, pflSnegopat)
profileRoot.createValue(pflActivateSearch, false, pflSnegopat)
var мДляВнешнихФайловОтображатьТолькоИмяФайла = profileRoot.getValue(pflOnlyNameForExtFiles);
var мИспользоватьСессии = profileRoot.getValue(pflUseSessions);
var activateSearchElement = profileRoot.getValue(pflActivateSearch);

sessionManager = null;
if (мИспользоватьСессии){
    loadSessionManager();
}

мФормаНастройки=null