仅在需要时如何自动添加工作区?


16

假设我正在使用4个工作空间,并且我偶然需要更多工作空间,是否有一个自动化的流程,或者,如果不可能的话,是否有一种简便的方法来偶然地添加更多工作空间(而不是安装Ubuntu tweak等)。



@JacobVlijm imgur.com/LaDYwGb Upvoted和接受,感谢非常好的答案
kernel_panic

相关:这篇旧帖子曾问“如何自动创建/删除更多工作区?” 有两个不同的答案。再者,与此处的较新答案相比,较旧的答案可能不太满意。
clearkimura 2015年

Answers:


14

自动设置工作区数量;根据需要添加和删除列和行

在backround脚本版本下,如果您输入了工作区矩阵的最后一列或最后一行,它将自动添加工作区。

它是这样工作的:

  1. 如果到达最后一列或最后一行,则会添加其他视口:

    在此处输入图片说明

  2. 如果您的工作区闲置了5到10秒钟,并且上面没有窗口,则其他工作区将被再次删除。但是,您将始终在下面保留额外的一行,并在当前视口的右边保留一列:

    在此处输入图片说明

剧本:

#!/usr/bin/env python3
import subprocess
import time
import math

# --- set default workspaces below (horizontally, vertically)
hsize = 2
vsize = 2
# --- set the maximum number of workspaces below
max_ws = 10

def set_workspaces(size, axis):
    subprocess.Popen([
        "dconf", "write", "/org/compiz/profiles/unity/plugins/core/"+axis,
                str(size)
                ])

def get_res():
    resdata = subprocess.check_output(["xrandr"]).decode("utf-8").split()
    curr = resdata.index("current")
    return (int(resdata[curr+1]), int(resdata[curr+3].replace(",", "")))

def wspace():
    try:
        sp = subprocess.check_output(["wmctrl", "-d"]).decode("utf-8").split()
        return ([int(n) for n in sp[3].split("x")],
                [int(n) for n in sp[5].split(",")])

    except subprocess.CalledProcessError:
        pass


def clean_up(curr_col, curr_row):
    try:
        w_list = [l.split() for l in subprocess.check_output(["wmctrl", "-lG"]).decode("utf-8").splitlines()]
        xpos = max([math.ceil((int(w[2])+span[1][0])/res[0]) for w in w_list])
        min_x = max(xpos, curr_col+1, hsize)
        if xpos >= hsize:
            set_workspaces(min_x, "hsize")
        else:
            set_workspaces(min_x, "hsize")
        ypos = max([math.ceil((int(w[3])+span[1][1])/res[1]) for w in w_list])
        min_y = max(ypos, curr_row+1, vsize)
        if ypos >= vsize:
            set_workspaces(min_y, "vsize")
        else:
            set_workspaces(min_y, "vsize")
    except subprocess.CalledProcessError:
        pass

res = get_res()
t = 0

while True:
    span = wspace()
    if span != None:
        cols = int(span[0][0]/res[0]); rows = int(span[0][1]/res[1])
        currcol = int((span[1][0]+res[0])/res[0])
        if all([currcol == cols, cols*rows < max_ws]):
            set_workspaces(cols+1, "hsize")
        currrow = int((span[1][1]+res[1])/res[1])
        if all([currrow == rows, cols*rows < max_ws]):
            set_workspaces(rows+1, "vsize")
        if t == 10:
            clean_up(currcol, currrow)
            t = 0
        else:
            t = t+1
        time.sleep(1)

如何使用

  1. 将以下脚本复制到一个空文件中,另存为 add_space.py
  2. 如果您喜欢其他设置(最大工作空间,默认矩阵,例如2x2),请在脚本的开头部分中编辑行:

    # --- set default workspaces below (horizontally, vertically)
    hsize = 2
    vsize = 2
    # --- set the maximum number of workspaces below
    max_ws = 10
    
  3. 通过以下命令对其进行测试:

    python3 /path/to/add_space.py
    
  4. 如果一切正常,请将其添加到启动应用程序中:Dash>启动应用程序>添加命令:

    /bin/bash -c "sleep 15 &&  python3 /path/to/add_space.py`
    

注意

与往常一样,该脚本非常“低耗”,并且不会给处理器增加任何明显的负载。

说明

下面的故事有点复杂,主要是对概念过程的解释,而不是编码。仅在有兴趣时阅读。

如何计算所需的工作空间(示例列)

输出的内容wmctrl -d如下:

0  * DG: 3360x2100  VP: 1680,1050  WA: 65,24 1615x1026  N/A

在输出中,VP: 1680,1050向我们提供有关跨越工作空间(所有视口矩阵)的位置的信息。仅当我们也具有屏幕的分辨率时,此信息才有用,因为例如1680,宽度可能是屏幕的两倍(不太可能,但仍然)或一倍。
幸运的是,我们可以从命令中解析出屏幕的分辨率xrandr

然后,如果我们知道屏幕的x尺寸为1680且当前为VP: 1680,1050,则知道我们位于工作区矩阵的第二列中。由于我们还知道了总矩阵(的大小DG: 3360x2100,也来自的输出wmctrl -d),因此我们知道当前矩阵包括两列(3360/1680),位于“最后”一列。

在此处输入图片说明

然后,脚本将发送一条指令,以通过以下命令在矩阵中添加一列:

dconf write /org/compiz/profiles/unity/plugins/core/hsize <current_viewport_column+1>

这就是原理。

如何计算要删除的工作空间(示例列)

脚本每10秒运行一次,并使用以下命令列出所有当前打开的窗口:

wmctrl -lG

这也为我们提供了有关窗口位置的信息,如下所示:

0x04604837  0 3425 24   1615 1026 jacob-System-Product-Name Niet-opgeslagen document 2 - gedit

在输出中,3425是窗口的x位置。但是,该图相对于当前工作空间(左侧)。要知道窗口在工作区矩阵中的绝对位置(x方向),我们必须添加当前视口信息的第一个数字(例如VP: 1680,1050,从的输出wmctrl -d)。
但是,出于简单起见,让我们假设我们在视口1,1(左上视口)上,因此窗口的相对位置等于其绝对位置。

由于屏幕的分辨率为1680,我们知道窗口在column上是3425/1680向上舍入的,因为之间的所有内容都在3360 and 5040矩阵的同一列上(分辨率在3到4倍之间)。为了进行正确的计算,我们使用math.ceil()python

由于脚本遵循规则,因此总是在右/下具有额外的工作区,因此我们需要将列数设置为以下最大值:

  • 当前工作空间柱+ 1
  • 最后与它的窗口列
  • 默认列数,如脚本开头所设置

脚本也是如此:)

以完全相同的过程管理行。


1
哇,好答案,已经赞成=)
AB

@AB谢谢!:)这是我喜欢做的事情:)
Jacob Vlijm 2015年

2
@kernel_panic哇,谢谢:)我实际上是想使其成为PPA,就像这样一个:askubuntu.com/a/608295/72216,而这个一个:askubuntu.com/a/560734/72216,最后(将来)将所有三个合并到一个“ workspace_tools”应用程序中,在其中可以将它们设置为选项或类似的内容:)。
Jacob Vlijm

1
@JacobVlijm为了简洁起见,我将从此处删除旧答案:对于感兴趣的人,在编辑历史记录中仍然可以看到...
Fabby 2015年

1
不用客气...您应该多去聊天室并经常出现毒素!:P
Fabby 2015年

6

从技术上讲,没有用于调整工作区大小的快捷方式,但是您可以使用下面的简单脚本并将其绑定到快捷方式。

  1. 在下面的脚本中,将其保存在 .local/share/applications文件夹中,或任何您喜欢的位置。
  2. 确保脚本可以通过以下方式执行 chmod 755 /path/to/script
  3. 将其绑定到系统设置->键盘->快捷方式->自定义快捷方式中的快捷方式

例如,我有以下设置:

在此处输入图片说明

该脚本已绑定到ShiftCtrlAltI。但是CtrlAltI也可以。我给出了脚本的完整路径,这是

/home/xieerqi/resize-workspaces.sh

这是它的外观:

在此处输入图片说明

脚本

#!/bin/bash
# Author : Serg Kolo
# Date: Sept 19, 2015
# Purpose: simple script to resize 
# unity workspaces
# Written for: http://askubuntu.com/q/676046/295286

HEIGHT=$(gsettings get org.compiz.core:/org/compiz/profiles/unity/plugins/core/ hsize)
WIDTH=$(gsettings get org.compiz.core:/org/compiz/profiles/unity/plugins/core/ vsize)
NEWSIZE=$(zenity --entry --text="Current desktop set-up $HEIGHT x $WIDTH. Enter new setup in HEIGHTxWIDTH format"  --width=250 | tr 'x' ' ' )

ARRAY=( $NEWSIZE )
[ -z ${ARRAY[1]}  ] && exit
    gsettings set org.compiz.core:/org/compiz/profiles/unity/plugins/core/ hsize ${ARRAY[0]}
    gsettings set org.compiz.core:/org/compiz/profiles/unity/plugins/core/ vsize ${ARRAY[1]}

使用非常简单,设置非常简单

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.